#pragma once
#include <iostream>
#include <string>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/wait.h>

#include "../comm/log.hpp"
#include "../comm/util.hpp"
namespace ns_runner
{
    using namespace ns_log;
    using namespace ns_util;
    class Runner
    {
    public:
        Runner() {}
        ~Runner() {}

    public:
        // 指明文件名即可，不需要带路径，因为默认创建的.exe都在./temp路径下
        /****************
         * 返回值 > 0 : 程序异常，退出时收到了信号，返回值就是对应的信号编号
         * 返回值 == 0: 正常运行完毕，结果保存到了对应的临时文件
         * 返回值 < 0 : 内部错误
         * 
        */
        static int Run(const std::string &file_name)
        {
            /*
             *  程序运行：
             *  1.代码跑完，结果正确
             *  2.代码跑完，结果不正确
             *  3.代码没跑完，异常(run不考虑该问题)
             *  结果正确与否，由测试用例决定
             *  我们只考虑：是否正确运行完毕

             *  我们必须知道可执行程序是谁
             *  一个程序在默认启动的时候
             *  标准输入：不处理
             *  标准输出：程序运行完成，输出结果是什么
             *  标准错误：运行时错误信息
            */

            std::string _execute = PathUtil::Exe(file_name);
            std::string _stdin = PathUtil::Stdin(file_name);
            std::string _stdout = PathUtil::Stdout(file_name);
            std::string _stderr = PathUtil::Stderr(file_name);

            umask(0);
            int _stdin_fd  = open(_stdin.c_str(), O_CREAT| O_WRONLY,0644);
            int _stdout_fd = open(_stdout.c_str(), O_CREAT| O_WRONLY,0644);
            int _stderr_fd = open(_stderr.c_str(), O_CREAT| O_WRONLY,0644);
            if(_stdin_fd < 0 || _stdout_fd < 0 || _stderr_fd < 0){
                LOG(ERROR) << "运行时打开标准文件错误" << "\n";
                return -1;//代表文件打开失败
            }

            pid_t pid = fork();
            if (pid < 0)
            {
                LOG(ERROR) << "运行时创建子进程失败" << "\n";
                close(_stdin_fd);
                close(_stdout_fd);
                close(_stderr_fd);
                return -2;//代表创建子进程失败
            }
            else if (pid == 0)
            {
                // 子进程执行新形成的程序
                dup2(_stdin_fd,0);
                dup2(_stdout_fd,1);
                dup2(_stderr_fd,2);

                execl(_execute.c_str()/*执行谁*/, _execute.c_str()/*如何执行*/, nullptr);
                exit(1);
            }
            else
            {
                close(_stdin_fd);
                close(_stdout_fd);
                close(_stderr_fd);
                int status = 0;
                waitpid(pid, &status, 0);
                //程序运行异常，一定是因为收到了信号！
                LOG(INFO) << "运行完毕，info: " << (status & 0x7F) << "\n";
                return status & 0x7F;
            }
        }
    };

}